package com.gdut.gulimall.order.config;

import org.springframework.amqp.core.Message;
import org.springframework.amqp.rabbit.connection.ConnectionFactory;
import org.springframework.amqp.rabbit.connection.CorrelationData;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.amqp.support.converter.Jackson2JsonMessageConverter;
import org.springframework.amqp.support.converter.MessageConverter;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;

import javax.annotation.PostConstruct;

/**
 * @author Drifiter
 * @create 2021-10-24 11:02
 */
@Configuration
public class MyRabbitConfig {

//    @Autowired
    RabbitTemplate rabbitTemplate;
//    public MyRabbitConfig(RabbitTemplate rabbitTemplate){
//        this.rabbitTemplate = rabbitTemplate;
//        initRabbitTemplate();
//    }
    @Primary
    @Bean
    public RabbitTemplate rabbitTemplate(ConnectionFactory connectionFactory){
        RabbitTemplate rabbitTemplate = new RabbitTemplate(connectionFactory);
        this.rabbitTemplate = rabbitTemplate;
        rabbitTemplate.setMessageConverter(messageConverter());
        initRabbitTemplate();
        return rabbitTemplate;
    }

    /**
     * 使用ISON序列化机制，进行消息转换
     * @return
     */
    @Bean
    MessageConverter messageConverter(){
        return new Jackson2JsonMessageConverter();
    }

    /**
     * 定制RabbitTemplate
     * 1、spring.rabbitmq.publisher-confirms=true2、设置确认回调
     *
     *  2、消息正确抵达队列进行回调
     * 1、spring.rabbitmq.pubLisher-returns=true
     * spring .rabbitmq.template.mandatory=true
     * 2、设置确认回调ReturnCaLLback
     * 3、消费端确认（保证每个消息被正确消费，此时才可以broker删除这个消息）。
     * 1、默认是自动确认的，只要消息接收到，客户端会自动确认，服务端就会移除这个消息
     * 间题:
     * 我们收到很多消息，自动回复给服务器ack，只有一个消息处理成功，宕机了。发生消息丢失:手动确认。|
     * 消费者手动确认模式。只要我们没有明确告诉NQ，货物被签收。没有Ack,
     * 消息就一直是unacked状态。即使consumer宕机。消息不会丢失，会重新变为Ready,
     * 2、如何签收:
     *          channel.basicAck(deliveryTag,false);签收;业务成功完成就应该签收
     *          channel.basicNack(deliveryTag,false,true);拒签;业务失败，拒签
     */
//    @PostConstruct
    public void initRabbitTemplate(){

        // 设置确认回调
        rabbitTemplate.setConfirmCallback(new RabbitTemplate.ConfirmCallback() {
            /**
             *  1、只要消息抵达Broker就ack=true
             * @param correlationData     当前消息的唯一关联数据（这个是消息的唯—id)
             * @param ack       消息是否成功收到
             * @param cause      失败的原因
             */
            @Override
            public void confirm(CorrelationData correlationData, boolean ack, String cause) {
                System.out.println("confirm....correlationData["+correlationData+"]==>ack=["+ack+"]==>cause==>["+cause+"]");
            }
        });

        rabbitTemplate.setReturnCallback(new RabbitTemplate.ReturnCallback() {
            /**
             * 只要消息没有投递给指定的队列,就触发这个失败回调
             * @param message    投递失败的消息详细信息
             * @param replyCode  回复的状态码
             * @param replyText  回复的文本内容
             * @param exchange   当时这个消息发给哪个交换机
             * @param routingKey   当时这个消息用哪个路由键
             */
            @Override
            public void returnedMessage(Message message, int replyCode, String replyText, String exchange, String routingKey) {
                System.out.println("Fail message["+message+"]==>replyCode["+replyCode+"]==>replyText["+replyText+"]==>exchange["+exchange+"]==>routingKey["+routingKey+"]");
            }
        });

    }

}
